from flask import abort
from flask import current_app, jsonify
from flask import g
from flask import render_template
from flask import request
from flask import session

from info import constants, db
from info.models import News, Comment, CommentLike, User
from info.response_code import RET
from info.utils.commons import user_login_data
from . import news_blu

# 关注与取消关注作者
# 请求路径: /news/followed_user
# 请求方式: POST
# 请求参数:user_id,action
# 返回值: errno, errmsg
@news_blu.route('/followed_user', methods=['POST'])
@user_login_data
def followed_user():
    """
    思路分析：
    1.判断用户是否有登陆
    2.获取参数
    3.校验参数，为空，类型校验
    4.通过编号查询作者，并判断作者是否存在
    5.根据操作类型，关注/取消操作111
    6.返回响应
    :return:
    """
    # 1.判断用户是否有登陆
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='用户未登录')

    # 2.获取参数
    author_id = request.json.get('user_id')
    action = request.json.get('action')

    # 3.校验参数，为空，类型校验
    if not all([author_id,action]):
        return jsonify(errno=RET.PARAMERR,errmsg='参数不全')

    if not action in ['follow','unfollow']:
        return jsonify(errno=RET.DATAERR,errmsg='操作类型异常')


    # 4.通过编号查询作者，并判断作者是否存在
    try:
        author = User.query.get(author_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='用户查询失败')

    if not author:
        return jsonify(errno=RET.NODATA,errmsg='用户不存在')

    # 5.根据操作类型，关注 / 取消操作
    if action == 'follow':
        # 判断是否有关注
        if not (g.user in author.followers):
            author.followers.append(g.user)
    else:
        # 判断是否有关注
        if g.user in author.followers:
            author.followers.remove(g.user)

    # 6.返回响应
    return jsonify(errno=RET.OK,errmsg='操作成功')






# 新闻详情展示
# 请求路径：/news/<int:news_id>
# 请求方式：get
# 请求参数：news_id
# 返回值：detail.html页面，用户data字典数据
@news_blu.route('/<int:news_id>')
@user_login_data
def news_details(news_id):

    # # 获取用户编号
    # user_id = session.get('user_id')
    #
    # # 查询用户对象
    # user = None
    # if user_id:
    #     try:
    #         user = User.query.get(user_id)
    #     except Exception as e:
    #         current_app.logger.error(e)

    # 根据编号，获取新闻对象
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='新闻查询失败')

    # 判断新闻是否存在，以后会对统一的404做处理
    if not news:
        abort(404)

    # 将新闻的点击量+1
    news.clicks += 1

    # 查询热门新闻列表，取前10条
    try:
        news_items = News.query.order_by(News.clicks.desc()).limit(constants.CLICK_RANK_MAX_NEWS)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='热门新闻查询失败')

    # 将新闻对象列表转成，字典列表
    clicks_news_list = []
    for item in news_items:
        clicks_news_list.append(item.to_dict())

    # 判断用户是否有收藏该新闻
    is_collected = False
    # 如果用户登陆，并且该新闻在用户的收藏列表中，说明该新闻收藏过
    if g.user and news in g.user.collection_news:
        is_collected = True

    # 获取该条新闻的所有评论内容
    try:
        comments = Comment.query.filter(Comment.news_id == news_id).order_by(Comment.create_time.desc()).all()
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='获取评论失败')

    # 记录点赞过的评论编号
    my_comment_like_ids = []
    if g.user:
        # 先获取到所有的评论列表编号（当前新闻）
        comments_ids = [comm.id for comm in comments]

        # for comm in comments:
        #     comment_ids.append(comm.id)
        # 获取到当前新闻，所有的评论的，所有赞对象列表
        # 条件1：找到当前新闻的所有赞（很多人） 条件2：过滤出某个人
        comment_like_list = CommentLike.query.filter(
            CommentLike.comment_id.in_(comments_ids)
            ,CommentLike.user_id == g.user.id).all()

        # 获取到用户对当前新闻，点赞过的评论编号
        my_comment_like_ids = [comment_like.comment_id for comment_like in comment_like_list]



    # 将评论对象列表转成，字典列表
    comment_list = []
    for comment in comments:
        comm_dict = comment.to_dict()
        # 假设对每条评论都没有点过赞
        comm_dict['is_like'] = False

        # 判断是否点赞
        if g.user and comment.id in my_comment_like_ids:
            comm_dict['is_like'] = True

        comment_list.append(comm_dict)


    # 判断用户是否关注过新闻的作者
    is_followed = False

    # 用户需要登陆，并且新闻要有作者
    if g.user and news.user:
        # 用户必须要在作者的粉丝列表中
        if g.user in news.user.followers:
            is_followed = True

    # 拼接数据，渲染到页面
    data = {
        'user_info':g.user.to_dict() if g.user else '',
        'news':news.to_dict(),
        'clicks_news_list':clicks_news_list,
        'is_collected':is_collected,
        'comments':comment_list,
        'is_followed':is_followed


    }
    return render_template('news/detail.html',data=data)


# 收藏/取消收藏
# 请求路径：/news/news_collect
# 请求方式：post
# 请求参数：news_ud,action,g.user
# 返回值：errno,errmsg

@news_blu.route('/news_collect', methods=['POST'])
@user_login_data
def news_collect():
    """
    思路分析：
    0.判断用户是否登陆
    1.取出参数
    2.校验参数，为空校验，参数类型校验
    3.根据编号查询新闻对象是否存在
    4.根据操作类型，收藏，取消收藏
    5.返回响应
    :return:
    """
    # 0.判断用户是否登陆
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='该用户未登陆')

    # 1.取出参数
    news_id = request.json.get('news_id')
    action = request.json.get('action')

    # 2.校验参数，为空校验，参数类型校验
    if not all([news_id,action]):
        return jsonify(errno=RET.NODATA,errmsg='参数不全')

    # 3.根据编号查询新闻对象是否存在
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='新闻查询异常')

    if not news:
        return jsonify(errno=RET.NODATA,errmsg='该新闻不存在')

    # 4.根据操作类型，收藏，取消收藏
    try:
        if action == 'collect':
            g.user.collection_news.append(news)
        else:
            g.user.collection_news.remove(news)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='操作')

    # 5.返回响应
    return jsonify(errno=RET.OK, errmsg='操作成功')


# 评论/回复评论
# 请求路径：/news/news_comment
# 请求方式：post
# 请求参数：news_id,comment,parent_id
# 返回值：errno,errmsg,评论字典

@news_blu.route('/news_comment', methods=['POST'])
@user_login_data
def news_comment():
    """
    思路分析：
    1.判断用户是否登陆
    2.获取参数
    3.校验，为空校验
    4.根据编号取出新闻对象，并判断是否存在
    5.创建评论对象，设置评论对象属性
    6.添加评论对象到数据库
    7.返回响应

    :return:
    """
    # 1.判断用户是否登陆
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='该用户未登陆')

    # 2.获取参数
    news_id = request.json.get('news_id')
    content = request.json.get('comment')
    parent_id = request.json.get('parent_id')

    # 3.校验，为空校验
    if not all([news_id,content]):
        return jsonify(errno=RET.PARAMERR,errmsg='参数不全')

    # 4.根据编号取出新闻对象，并判断是否存在
    try:
        news = News.query.get(news_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='新闻查询失败')

    if not news:
        return jsonify(errno=RET.NODATA,errmsg='该新闻不存在')

    # 5.创建评论对象，设置评论对象属性
    comment = Comment()
    comment.user_id = g.user.id
    comment.news_id = news.id
    comment.content = content

    if parent_id:
        comment.parent_id = parent_id

    # 6.添加评论对象到数据库
    try:
        db.session.add(comment)
        db.session.commit()
    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=RET.DBERR,errmsg='添加评论失败')

    # 7.返回响应
    return jsonify(errno=RET.OK,errmsg='添加评论成功',data=comment.to_dict())

# 点赞/取消点赞
# 请求路径: /news/comment_like
# 请求方式: POST
# 请求参数:news_id,comment_id,action,g.user
# 返回值: errno,errmsg

@news_blu.route('/comment_like', methods=['POST'])
@user_login_data
def user_comment_like():
    """
    思路分析：
    1.判断用户是否登陆
    2.获取参数
    3.校验参数，为空校验，类型校验
    4.根据评论编号查询评论对象
    5.判断评论对象是否存在
    6.根据操作类型，点赞，取消
    7.返回响应
    :return:
    """
    # 1.判断用户是否登陆
    if not g.user:
        return jsonify(errno=RET.NODATA,errmsg='用户未登陆')

    # 2.获取参数
    comment_id = request.json.get('comment_id')
    action = request.json.get('action')

    # 3.校验参数，为空校验，类型校验
    if not all([comment_id,action]):
        return jsonify(errno=RET.PARAMERR,errmsg='参数不全')

    if not action in ('add','remove'):
        return jsonify(errno=RET.DATAERR,errmsg='操作类型有误')

    # 4.根据评论编号查询评论对象
    try:
        comment = Comment.query.get(comment_id)
    except Exception as e:
        current_app.logger.error(e)
        return jsonify(errno=RET.DBERR,errmsg='评论查询失败')

    # 5.判断评论对象是否存在
    if not comment:
        return jsonify(errno=RET.DBERR,errmsg='该评论不存在')

    # 6.根据操作类型，点赞，取消
    try:

        if action == 'add':
            # 查询用户是否对该评论点过赞
            comment_like = CommentLike.query.filter(
                CommentLike.comment_id == comment_id,
            CommentLike.user_id == g.user.id).first()
            if not comment_like:
                # 创建点赞对象，设置属性值
                comment_like = CommentLike()
                comment_like.comment_id = comment_id
                comment_like.user_id = g.user.id

                # 将当前评论的点赞数量+1
                comment.like_count += 1

                # 添加到数据库（要加try）

                db.session.add(comment_like)
                db.session.commit()

        else:
            # 查询用户是否对该评论点过赞
            comment_like = CommentLike.query.filter(
                CommentLike.comment_id == comment_id,
            CommentLike.user_id == g.user.id).first()
            if comment_like:
                db.session.delete(comment_like)

                # 将当前评论的点赞数量-1
                comment.like_count -= 1

                db.session.commit()

    except Exception as e:
        current_app.logger.error(e)
        db.session.rollback()
        return jsonify(errno=RET.DBERR,errmsg='点赞失败')

    # 7.返回响应
    return jsonify(errno=RET.OK,errmsg='点赞成功')



















